iT邦幫忙

2017 iT 邦幫忙鐵人賽
DAY 28
0
自我挑戰組

我的語言自學接力系列 第 28

D28:[Java]凍結執行緒 join

  • 分享至 

  • xImage
  •  

在[Java]執行緒生命週期(介紹)中,被凍結的狀態提到的 sleep() 已在[Java]凍結執行緒 sleep 說明了用法;本篇將說明 join() 的用法。

在[Java]凍結執行緒 sleep 中的範例,兩個執行緒(apple,pen)可說是同時啟動的,那如果再啟動的下方再加上一行印出 main() 結束 的程式碼

apple.show();
pen.show();
System.out.println("main() 結束");

完整程式碼:

class multi_Test
{
   private String name;
   public multi_Test(String str)
   {
      name=str;
   }
   public void show()
   {
      for(int i=0;i<3;i++)
      {
         try
         {
            sleep((int)(1000*Math.random()));
         }
         catch(InterruptedException e){}
         System.out.println(name);
      }
   }
}

public class multi_thread_3
{
   public static void main(String args[])
   {
      multi_Test apple=new multi_Test("apple");
      multi_Test pen=new multi_Test("pen");
      apple.start();
      pen.start();
      System.out.println("main() 結束");
   }
}

當我們執行的結果會發現,會先印出「main() 結束」。

記得 main() 本身其實也是個執行緒,一樣是看誰先搶到資源。
但因為僅印出字串,不需要經過執行緒的啟動,因此通常較先搶到資源 先執行。

這麼一來若是先印出「 main() 結束」 就不是我們想要的目的;
我們想先apple 執行緒完成後再執行pen 執行緒,
等到兩個執行緒都結束才正式main() 結束(因此印出字串)。

想要達到這樣的結果,就必須使用到 join() 來處理執行緒的排程。

在[Java]執行緒生命週期(介紹)中提到的觀念:

執行緒與另一個執行緒 join() 一起時:
當有其他執行緒呼叫 join(),原來正執行的執行緒(或程式碼)會先暫停

join() 必須寫在 try-catch 區塊內,因此我們可以將上述錯誤的程式這樣改寫:

當 apple 執行緒啟動 便進到try 後,apple執行緒結束後就會往下執行 啟動pen執行緒,
當pen 執行緒結束後 才會再往下執行 印出字串的程式。

apple.start();
try
{
   apple.join();
   pen.start();
   pen.join();
}
catch(){}
System.out.println("main() 結束");

在啟動 apple 執行緒後會先停留在 apple.join() 等待該執行緒完成,才會執行下一行程式(啟動pen執行緒);
同理 pen 執行緒會停留在 pen.join() ,等到執行緒結束後就會將字串「main() 結束」字串印出。


還有一點要注意的是, join() 會拋出 InterruptedException 例外
因此才必須將 join() 寫在 try-catch 區塊,而catch 要捕捉 InterruptedException 例外。

apple.start();
try
{
   apple.join();
   pen.start();
   pen.join();
}
catch(InterruptedException e){}
System.out.println("main() 結束");

最後附上完整正確的程式碼:

class multi_Test
{
   private String name;
   public multi_Test(String str)
   {
      name=str;
   }
   public void show()
   {
      for(int i=0;i<3;i++)
      {
         try
         {
            sleep((int)(1000*Math.random()));
         }
         catch(InterruptedException e){}
         System.out.println(name);
      }
   }
}

public class multi_thread_3
{
   public static void main(String args[])
   {
      multi_Test apple=new multi_Test("apple");
      multi_Test pen=new multi_Test("pen");
      apple.start();
      try
      {
         apple.join();
         pen.start();
         pen.join();
      }  
      catch(InterruptedException e){}
      System.out.println("main() 結束");
   }
}

如果您喜歡我的文章,可以至 4xDevJava 學習系列閱讀更多 Java 文章。


上一篇
D27:[Java]凍結執行緒 sleep
下一篇
D29:[Java]執行緒的順序
系列文
我的語言自學接力31
圖片
  直播研討會
圖片
{{ item.channelVendor }} {{ item.webinarstarted }} |
{{ formatDate(item.duration) }}
直播中

尚未有邦友留言

立即登入留言